2.4 Web components

2.4.1 Introducción:

Requisitos

  1. Navegador compatible.
  2. Conocimientos básicos de HTML, CSS, JS.
  3. Familiaridad con módulos JS y funciones de flecha.

Lo que aprenderás

  1. Elementos personalizados (Custom Elements).
  2. Plantillas (Templates).
  3. Shadow DOM.

2.4.2 Setup:

To get started, let's create a basic HTML page:

<html>
  <body>
    <h1>Hello world!</h1>
  </body>
</html>

2.4.3 Custom Elements

2.4.4 Customizing our element

<cool-heading color="red">
  <h1>Hello world!</h1>
</cool-heading>
<script>
  class CoolHeading extends HTMLElement {
    constructor() {
      super();
    }

    connectedCallback() {
      this.style.color = this.getAttribute('color');
    }

    static get observedAttributes() {
      return ['color'];
    }

    attributeChangedCallback(name, oldValue, newValue) {
      this.style.color = newValue;
    }
  }
  customElements.define('cool-heading', CoolHeading);
  // <cool-heading color="red">Hello world!</cool-heading>
  // <cool-heading color="blue">Hello world!</cool-heading>
  // Usaremos el atributo color para cambiar el color del texto.

  setTimeout(() => {
    document.querySelector('cool-heading').setAttribute('color', 'blue');
  }, 2000);
</script>

2.5 Templating


2.6 Shadow DOM


2.7 Web Components en la práctica

1. YouTube

2. GitHub

3. Twitter

4. Elemento de Video

5.1 Basic Dialog Custom Element

Para crear un diálogo básico, añade un botón de apertura y un botón de cierre al contenido del diálogo. Y luego añade lógica para abrir y cerrar el diálogo.

<!DOCTYPE html>
<html>
  <body>
    <custom-dialog>
      <button class="dialog-btn">Open</button>
      <dialog open>
        <button class="close-btn">🗙</button>
        <h1>Dialog</h1>
        <p>This is a dialog.</p>
      </dialog>
    </custom-dialog>
  </body>
</html>

5.1.1 Custom Dialog Element

Para crear un elemento personalizado, extiende HTMLElement y añade un connectedCallback() para manejar la lógica de apertura y cierre del diálogo.

<!DOCTYPE html>
<html>
  <body>
    <custom-dialog>
      <button class="dialog-btn">Open</button>
      <dialog open>
        <button class="close-btn">🗙</button>
        <h1>Dialog</h1>
        <p>This is a dialog.</p>
      </dialog>
    </custom-dialog>

    <style>
      .close-btn {
        background-color: #f44336;
        color: white;
        border: none;
        cursor: pointer;
        padding: 10px 20px;
        position: absolute;
        right: 0;
        top: 0;
      }
      dialog {
        background-color: #ffffff;
        border-radius: 4px;
        box-shadow: 0 0 10px #00000061;
        width: 400px;
        padding: 20px;
        position: relative;
        text-align: center;
      }
      dialog::backdrop {
        background-color: #00000061;

      }
    </style>

    <script>
      class CustomDialog extends HTMLElement {
        constructor() {
          super();
        }
        connectedCallback() {
          // TODO: Add dialog open/close logic
          this.
        }
      }
      customElements.define('custom-dialog', CustomDialog);
    </script>
  </body>
</html>

5.1.2 Styling the Dialog

Para añadir estilos al diálogo, utiliza CSS en línea o en un bloque <style> en la página.

6. <Dialog> + <Form> -> (dispatchEvent)

Para crear un diálogo con un formulario, añade un formulario al contenido del diálogo. Luego, añade lógica para enviar los datos del formulario al componente principal.

dispatchEvent:

<custom-dialog>
  <button class="dialog-btn">Open</button>
  <dialog>
    <button class="close-btn">🗙</button>
    <h1>Dialog</h1>
    <form>
      <label for="input">Input</label>
      <input id="input" type="text" />
      <button type="submit">Submit</button>
    </form>
  </dialog>
</custom-dialog>

<style>
  .close-btn {
    background-color: #f44336;
    color: white;
    border: none;
    cursor: pointer;
    padding: 10px 20px;
    position: absolute;
    right: 0;
    top: 0;
  }
  dialog {
    background-color: #ffffff;
    border-radius: 4px;
    box-shadow: 0 0 10px #00000061;
    width: 400px;
    padding: 20px;
    position: relative;
    text-align: center;
  }
</style>

<script>
  class CustomDialog extends HTMLElement {
    constructor() {
      super();
    }

    submit(event) {
      event.preventDefault();
      const input = this.querySelector('input');
      // Send data outside
      this.dispatchEvent(new CustomEvent('custom-submit', {
        detail: { input: input.value }
      }));
      this.close();
    }

    close() {
      const dialog = this.querySelector('dialog');
      dialog.close();
    }

    connectedCallback() {
      const dialogBtn = this.querySelector('.dialog-btn');
      const dialog = this.querySelector('dialog');
      const closeBtn = this.querySelector('.close-btn');
      const form = this.querySelector('form');
      const input = this.querySelector('input');

      // TODO listeners
      dialogBtn.addEventListener('click', () => {
        dialog.showModal();
      });
      closeBtn.addEventListener('click', () => this.close());

      // TODO dispatchEvent
      form.addEventListener('submit', event => this.submit(event))
    }
  }

  customElements.define('custom-dialog', CustomDialog);

  // TODO usage dispatchEvent and addEventListener
  const customDialog = document.querySelector('custom-dialog');
  // Listen data from dialog
  customDialog.addEventListener('custom-submit', (event) => {
    console.log({detail: event.detail}); // 'value'
  });
</script>

7. Relative Time Web Component

Para crear un componente de tiempo relativo, añade un elemento de tiempo con un atributo datetime al contenido del componente. Luego, añade lógica para mostrar el tiempo relativo en lugar de la fecha y hora absolutas.

<relative-time time="Wed Oct 16 2023 09:17:41 GMT+0200"/>

<script>
  class RelativeTime extends HTMLElement {
    constructor() {
      super();
    }
    /*
    connectedCallback() {
      this.render();
      setInterval(() => {
        this.render();
      }, 1000)
    }
    */
    connectedCallback() {
      const time = new Date(this.getAttribute('time')).getTime();
      const now = Date.now();

      console.log({
        time,
        now,
        seconds: (now - time) / 1000,
        minutes: (now - time) / (1000 * 60),
        hours: Math.floor((now - time) / (1000 * 60 * 60))
      })

      const diff = now - time;
      const seconds = Math.floor(diff / 1000);
      const minutes = Math.floor(diff / (1000 * 60));
      const hours = Math.floor(diff / (1000 * 60 * 60));
      const days = Math.floor(hours / 24);
      const months = Math.floor(days / 30);
      const years = Math.floor(months / 12);

      let aux = '...';
      if (months >= 12) {
        aux = `Hace ${years} año${years > 1 ? 's' : ''}`
      } else if (days > 30 && months >= 1) {
        aux = `Hace ${months} mes${months > 1 ? 'es' : ''}`
      } else if (days >= 1) {
        aux = `Hace ${days} día${days > 1 ? 's' : ''}`
      } else if (hours >= 1) {
        aux = `Hace ${hours} hora${hours > 1 ? 's' : ''}`
      } else if (minutes >= 1) {
        aux = `Hace ${minutes} minuto${minutes > 1 ? 's' : ''}`
      } else if (seconds >= 1) {
        aux = `Hace ${seconds} segundo${seconds > 1 ? 's' : ''}`
      }

      this.textContent = aux;
    } 
  }
  customElements.define('relative-time', RelativeTime);
</script>

Práctica 3: Crear una página de noticias